home *** CD-ROM | disk | FTP | other *** search
/ Amiga Plus 2000 #4 / Amiga Plus CD - 2000 - No. 4.iso / Vollversion / CamD / development / examples / drum / drum.c < prev    next >
C/C++ Source or Header  |  2000-05-15  |  61KB  |  1,871 lines

  1. /* ============================================================================ *
  2.                  drum.c -- a simple drum machine using CAMD and RealTime
  3.                                       By Talin.
  4.                                ©1991 The Dreamers Guild
  5.  * ============================================================================ */
  6.  
  7. #define MCLOCKS_PER_QNOTE   480
  8.  
  9. /* ============================================================================ *
  10.                                         Notes
  11.  * ============================================================================ */
  12.  
  13. #if 0
  14.     What's next?
  15.         -- program gadget for setup. Init strings? Standard init strings?
  16.             -- a UI to compose a set of MIDI strings...?
  17.         -- mute and solo gadgets. (Do as bevel boxes?)
  18.         -- allow numpad keys to also work
  19.         -- disallow key repeats.
  20.         -- pattern length and resolution.
  21.         -- do maestro calculations time correctly.
  22.         -- solve maestro locating problem.
  23.         -- seperate task to handle the actual timing chores
  24.         -- recording as we go.
  25.         -- add in toggle buttons (boopsi) for mute. (or just "roll-yer-own??)
  26.         -- add in conductor setting and output link setting lists
  27.                 (use transport code...)
  28.         -- hook up individual menu items.
  29.  
  30.     Menus:
  31.         Project
  32.             Load
  33.             Save
  34.         ?   Save Settings?
  35.             Quit
  36.             About
  37.         Edit
  38.             ?? Undo ??
  39.             Erase Pattern
  40.             Reverse Pattern
  41.         Play
  42.             Whole Song
  43.             --------------
  44.             Set Conductor...
  45.             Set MIDI Output Port...
  46.         Pattern
  47.             Set Pattern Length...
  48.             Set Drum Resolution >>  (Quarter note, etc...)
  49.  
  50.     Other ideas:
  51.         -- a pattern arrangement dialogue...
  52.         -- play backwards.
  53.         -- ARexx port?
  54.  
  55.     WA_MenuHelp
  56.     WA_NotifyDepth
  57. #endif
  58.  
  59. /* ============================================================================ *
  60.                                        Includes
  61.  * ============================================================================ */
  62.  
  63. #include "std_headers.h"
  64. #include <intuition/icclass.h>
  65. #include <libraries/iffparse.h>
  66. #include <libraries/asl.h>
  67. #include <libraries/realtime.h>
  68. #include <midi/mididefs.h>
  69. #include <midi/camd.h>
  70. #include <clib/asl_protos.h>
  71. #include <clib/camd_protos.h>
  72. #include <clib/realtime_protos.h>
  73. #include <clib/alib_protos.h>
  74. #include <pragmas/realtime_pragmas.h>
  75. #include <pragmas/camd_pragmas.h>
  76. #include <stdlib.h>
  77. #include <math.h>
  78.  
  79. #include "arrows.h"                             /* increment arrows             */
  80.  
  81. /* ============================================================================ *
  82.                                       Prototypes
  83.  * ============================================================================ */
  84.  
  85. struct DrumColumn;
  86. void draw_titlebar_scale( void );
  87. void render_column( int column, int highlight );
  88. void play_drum_note( struct DrumColumn *dc, BOOL note_on );
  89. void preview_row( int row );
  90. void step_play( int dir );
  91.  
  92. BOOL save_pattern( void );
  93.  
  94. void calc_tempo( void );
  95. void calc_pattern_ticks( void );
  96. void locate_to_tick ( long tick_count );
  97.  
  98. struct Gadget *make_arrow( struct NewGadget *ng, struct Gadget *prev, int min, int max, int current );
  99. ULONG AddTailObject( Object *o, struct List *list);
  100. void DisposeObjectList( struct List *l );
  101. void shadow_drag( int col1, int col2 );
  102.  
  103. extern Class *initArrowButtonClass(void);
  104. void freeArrowButtonClass(Class *sclass);
  105.  
  106. /* ============================================================================ *
  107.                                        Defines
  108.  * ============================================================================ */
  109.  
  110. #define clamp(a,b,c)    min( max( a, b ), c )   /* constrain b between a and c  */
  111.  
  112. #define APPNAME         "Drum Machine"
  113. #define DRUM_ROWS       10
  114. #define ROW_HEIGHT      14
  115. #define MAX_MATRIX_WIDTH 400
  116. #define MAX_COLUMNS     64
  117. #define MAX_PATTERNS    128
  118. #define MATRIX_X        224
  119.  
  120. enum gadget_ids {
  121.     ID_MUTE = 1,                                /* mute button per row          */
  122.     ID_SOLO,                                    /* solo button per row          */
  123.     ID_PNAME,                                   /* name string per row          */
  124.     ID_KEY,                                     /* key to play -- per row       */
  125.     ID_KEYINC,                                  /* key increment arrows         */
  126.     ID_HIT,                                     /* hit drum now...              */
  127.     ID_FILE,                                    /* filename                     */
  128.     ID_PLAY,                                    /* play button                  */
  129.     ID_STOP,                                    /* stop button                  */
  130.     ID_STEPFWD,                                 /* step fwd button              */
  131.     ID_STEPBACK,                                /* step backwd button           */
  132.     ID_CHANNEL,                                 /* channel slider               */
  133.     ID_CHANNELINC,
  134.     ID_TEMPO,                                   /* tempo slider                 */
  135.     ID_VELOCITY,                                /* velocity slider              */
  136. };
  137.  
  138.     /*  Normally, the columns of the drum grid are equal to a sixteenth note
  139.         in duration, however this can be changed. Here are the values that
  140.         the "drum resolution" value can take.
  141.     */
  142.  
  143. enum drum_res {
  144.     DRES_NOTE1=0,                               /* whole note                   */
  145.     DRES_NOTE2,                                 /* half note                    */
  146.     DRES_NOTE4,                                 /* quarter note                 */
  147.     DRES_NOTE8,                                 /* eighth note                  */
  148.     DRES_NOTE16,                                /* etc. */
  149.     DRES_NOTE32,
  150.     DRES_NOTE64,
  151.     DRES_NOTE128,
  152. };
  153.  
  154.     /*  Defines a single column of the drum pattern */
  155.  
  156. struct DrumColumn {
  157.     UBYTE               note[ DRUM_ROWS ];
  158. };
  159.  
  160.     /*  Defines the setup information for the drum kit */
  161.  
  162. struct DrumSetEntry {
  163.     char                name[20];
  164.     UBYTE               key;
  165.     UBYTE               flags;
  166. };
  167.  
  168. typedef struct DrumSetEntry DrumKit[ DRUM_ROWS ];
  169.  
  170. #define DRUMF_MUTE      (1<<0)
  171.  
  172. struct DrumPattern {
  173.     UWORD               dpNumber;               /* pattern number               */
  174.     UWORD               dpColumns,              /* pattern columns (global now) */
  175.                         dpRows;                 /* pattern rows (constant now)  */
  176.     DrumKit             dpSetup;                /* setup for pattern            */
  177.     struct DrumColumn   dpGrid[ MAX_COLUMNS ];  /* actual pattern data          */
  178. };
  179.  
  180. DrumKit                 ReadKit;
  181.  
  182. /* ============================================================================ *
  183.                                        Globals
  184.  * ============================================================================ */
  185.  
  186. struct Library          *IntuitionBase,
  187.                         *GadToolsBase,
  188.                         *AslBase,
  189.                         *GfxBase,
  190.                         *UtilityBase,
  191.                         *CamdBase,
  192.                         *RealTimeBase,
  193.                         *IFFParseBase;
  194.  
  195. extern struct Library   *SysBase,
  196.                         *DOSBase;
  197.  
  198. struct Window           *window;
  199. struct RastPort         *rp;
  200. struct VisualInfo       *vi;
  201. struct DrawInfo         *dri;
  202. UWORD                   *dri_pens;
  203. struct FileRequester    *filereq;
  204.  
  205. BOOL                    running = TRUE;
  206.  
  207. struct MidiNode         *midi;
  208. struct MidiLink         *out_link;
  209.  
  210. struct Gadget           *context_gadget,
  211.                         *first_gadget,
  212.                         *last_gadget,
  213.                         *first_arrow,
  214.                         *arrow_gadget;
  215.  
  216. struct Gadget           *key_gadgets[ DRUM_ROWS ],
  217.                         *name_gadgets[ DRUM_ROWS ],
  218.                         *keyinc_gadgets[ DRUM_ROWS ],
  219.                         *channel_string,
  220.                         *channel_arrow;
  221.  
  222. BOOL                    tempo_drag;
  223.  
  224. WORD                    drag_state,
  225.                         drag_row,
  226.                         drag_column,
  227.                         drag_anchor_column;
  228.  
  229. #define DRAG_SET        1
  230. #define DRAG_RESET      2
  231.  
  232. struct MinList          arrow_list;
  233.  
  234. struct TextAttr         topaz8 = { (STRPTR)"topaz.font", 8, FS_NORMAL, 0x0 };
  235.  
  236. struct BitMap           *off_bm;
  237. struct RastPort         off_rp,
  238.                         off_shine_rp,
  239.                         off_shade_rp;
  240.  
  241. void                    do_quit( struct IntuiMessage *msg );
  242. void                    do_about( struct IntuiMessage *msg );
  243. void                    do_savepattern( struct IntuiMessage *msg );
  244. void                    do_loadpattern( struct IntuiMessage *msg );
  245. void                    do_clear( struct IntuiMessage *imsg );
  246.  
  247. struct NewMenu menu_list[] = {
  248.     {   NM_TITLE,   "Project",          NULL,           NULL,   0,  NULL },
  249.     {   NM_ITEM,    "Load Pattern...",  (UBYTE *)"L",   NULL,   0,  do_loadpattern },
  250.     {   NM_ITEM,    "Load Song...",     (UBYTE *)"O",   ITEMENABLED,    0,  NULL },
  251.     {   NM_ITEM,    "Import from MIDI File...",NULL,    ITEMENABLED,0,NULL },
  252.     {   NM_ITEM,    NM_BARLABEL,        NULL,           NULL,   0L },
  253.     {   NM_ITEM,    "Save Pattern...",  (UBYTE *)"S",   NULL,   0,  do_savepattern },
  254.     {   NM_ITEM,    "Save Song...",     (UBYTE *)"W",   ITEMENABLED,0,NULL },
  255.     {   NM_ITEM,    "Export to MIDI File...",NULL,      ITEMENABLED,0,NULL },
  256.     {   NM_ITEM,    NM_BARLABEL,        NULL,           NULL,   0L },
  257.     {   NM_ITEM,    "About...",         (UBYTE *)"?",   NULL,   0,  do_about },
  258.     {   NM_ITEM,    "Quit",             (UBYTE *)"Q",   NULL,   0,  do_quit },
  259.     {   NM_TITLE,   "Edit",             NULL,           NULL,   0,  NULL },
  260.     {   NM_ITEM,    "Copy Pattern",     (UBYTE *)"C",   ITEMENABLED,    0,  NULL },
  261.     {   NM_ITEM,    "Paste Pattern",    (UBYTE *)"V",   ITEMENABLED,    0,  NULL },
  262.     {   NM_ITEM,    "Erase Pattern",    (UBYTE *)"E",   NULL,   0,  do_clear },
  263.     {   NM_ITEM,    NM_BARLABEL,        NULL,           NULL,   0L },
  264.     {   NM_ITEM,    "Reverse Pattern",  (UBYTE *)"R",   ITEMENABLED,    0,  NULL },
  265.     {   NM_TITLE,   "Playback",         NULL,           NULL,           0,  NULL },
  266.     {   NM_ITEM,    "Play",             NULL,           ITEMENABLED,    0,  NULL },
  267.     {   NM_SUB,     "Whole Song",       (UBYTE *)"H",   CHECKIT | CHECKED,  0, NULL },
  268.     {   NM_SUB,     "Pattern",          (UBYTE *)"P",   CHECKIT,            0, NULL },
  269.     {   NM_ITEM,    NM_BARLABEL,        NULL,           NULL,   0L },
  270.     {   NM_ITEM,    "Set Conductor...", (UBYTE *)"T",   ITEMENABLED,    0,  NULL },
  271.     {   NM_ITEM,    "Set MIDI Output...",(UBYTE *)"M",  ITEMENABLED,    0,  NULL },
  272.     {   NM_TITLE,   "Pattern",          NULL,           NULL,   0,  NULL },
  273.     {   NM_ITEM,    "Set Pattern Length...",(UBYTE *)"G",ITEMENABLED,   0, NULL },
  274.     {   NM_ITEM,    "Set Resolution...",NULL,           NULL,           0, NULL },
  275.     {   NM_END,     NULL,               NULL,           0,      0,  NULL }
  276. };
  277.  
  278. struct Menu             *menu_strip;
  279.  
  280.     /* drum matrix variables */
  281.  
  282. WORD                    matrix_top,
  283.                         matrix_bottom,
  284.                         matrix_height,
  285.                         matrix_y_org;
  286.  
  287. WORD                    total_columns = 32;
  288.                         column_width,
  289.                         matrix_width;
  290.  
  291. struct DrumPattern      *pattern_table[ MAX_PATTERNS ];
  292. struct DrumPattern      base_pattern,
  293.                         *current_pattern = &base_pattern;
  294.  
  295. WORD                    solo_drum = -1;
  296.  
  297. WORD                    current_velocity = 64,
  298.                         current_tempo = 100,
  299.                         current_channel = 5;
  300.  
  301. Class                   *aclass = NULL;
  302.  
  303.     /* playback variables */
  304.  
  305. struct Player           *drum_player;
  306.  
  307. WORD                    playback_column = 0;
  308.  
  309. WORD                    clock_state;
  310. struct Task             *main_task;
  311.  
  312.     /*  Values which are set before playing begins */
  313.  
  314. enum drum_res           drum_resolution = DRES_NOTE16; /* size of each column       */
  315.  
  316. UWORD                   mclocks_per_column;     /* metric clocks per column     */
  317. LONG                    pattern_mclocks;        /* metric clocks per pattern    */
  318.  
  319.     /*  Values which can change during play */
  320.  
  321. UWORD                   tempo_rate;             /* ratio of mtime / real time   */
  322.  
  323.     /*  Values which are continuously incremented during play */
  324.  
  325. LONG                    current_pattern_start,  /* start time of current pattern */
  326.                         current_pattern_end;    /* start time of next pattern   */
  327.  
  328. LONG                    clock_pos,              /* clock position               */
  329.                         mclock_accumulator,     /* metric time accumulator      */
  330.                         mclock_pos;             /* metric time position         */
  331.  
  332. LONG                    pattern_pos;            /* score position, in patterns  */
  333. UWORD                   column_pos;             /* column position in pattern   */
  334.  
  335.  
  336. #define ASM __asm
  337. #define REG(x) register __## x
  338.  
  339. extern ULONG ASM TimeServer (REG(a1)struct pmTime *msg, REG(a2)struct Player *pi);
  340.  
  341. struct Hook myHook = {
  342.     { NULL, NULL },
  343.     TimeServer,
  344. };
  345.  
  346. /* ============================================================================ *
  347.                                     Glue Functions
  348.  * ============================================================================ */
  349.  
  350. struct MidiNode *CreateMidi( Tag tag, ... )
  351. {   return CreateMidiA( (struct TagItem *)&tag );
  352. }
  353.  
  354. BOOL SetMidiAttrs( struct MidiNode *mi, Tag tag, ... )
  355. {   return SetMidiAttrsA( mi, (struct TagItem *)&tag );
  356. }
  357.  
  358. struct MidiLink *AddMidiLink( struct MidiNode *mi, LONG type, Tag tag, ... )
  359. {   return AddMidiLinkA( mi, type, (struct TagItem *)&tag );
  360. }
  361.  
  362. BOOL SetMidiLinkAttrs( struct MidiLink *mi, Tag tag, ... )
  363. {   return SetMidiLinkAttrsA( mi, (struct TagItem *)&tag );
  364. }
  365.  
  366. struct Player *CreatePlayer( Tag tag, ... )
  367. {   return CreatePlayerA( (struct TagItem *)&tag );
  368. }
  369.  
  370. BOOL SetPlayerAttrs( struct Player *pi, Tag tag, ... )
  371. {   return SetPlayerAttrsA( pi, (struct TagItem *)&tag );
  372. }
  373.  
  374. /* ============================================================================ *
  375.                                     Error Function
  376.  * ============================================================================ */
  377.  
  378. LONG ErrorF( struct Window *w, UBYTE *TextFormat, ... )
  379. {   struct EasyStruct es;
  380.  
  381.     es.es_StructSize = sizeof(struct EasyStruct);
  382.     es.es_Flags = 0;
  383.     es.es_Title = APPNAME " Error Request";
  384.     es.es_TextFormat = TextFormat;
  385.     es.es_GadgetFormat = "Continue";
  386.  
  387.     return( EasyRequestArgs( w, &es, NULL, (APTR)( (&TextFormat) + 1) ));
  388. }
  389.  
  390. struct Library *open_lib( char *name )
  391. {   struct Library          *l;
  392.  
  393.     unless ( l = OpenLibrary( name, 0 ))
  394.     {   ErrorF( NULL, "Can't open library:\n%s", name);
  395.     }
  396.     return l;
  397. }
  398.  
  399. /* ============================================================================ *
  400.                                      main routine
  401.  * ============================================================================ */
  402.  
  403. void main( int argc, char *argv[] )
  404. {   struct NewGadget    ng;
  405.     short               row, column;
  406.     BOOL                gadgets_added = FALSE;
  407.     WORD                last_qualifier = 0;
  408.  
  409.     main_task = FindTask( 0 );
  410.  
  411.     calc_pattern_ticks();
  412.     calc_tempo();
  413.     base_pattern.dpColumns = total_columns;
  414.     base_pattern.dpRows = DRUM_ROWS;
  415.     base_pattern.dpNumber = 0;
  416.  
  417.     NewList( (struct List *)&arrow_list );
  418.  
  419.         /* open libraries */
  420.  
  421.     unless (IntuitionBase = open_lib( "intuition.library" )) LEAVE;
  422.     unless (GadToolsBase = open_lib( "gadtools.library" )) LEAVE;
  423.     unless (AslBase = open_lib( "asl.library" )) LEAVE;
  424.     unless (GfxBase = open_lib( "graphics.library" )) LEAVE;
  425.     unless (UtilityBase = open_lib( "utility.library" )) LEAVE;
  426.     unless (CamdBase = open_lib( "camd.library" )) LEAVE;
  427.     unless (RealTimeBase = open_lib( "realtime.library" )) LEAVE;
  428.     unless (IFFParseBase = OpenLibrary( "iffparse.library", 0L )) LEAVE;
  429.  
  430.     aclass = initArrowButtonClass();            /* arrow buttons                */
  431.  
  432.         /* create the MIDI stuff */
  433.  
  434.     unless (midi = CreateMidi(
  435.         MIDI_Name, "VU Meters",
  436.         MIDI_ErrFilter, CMEF_All,
  437.         TAG_DONE ))
  438.     {   ErrorF( window, "Unable to create MIDI Node." );
  439.         LEAVE;
  440.     }
  441.  
  442.     unless (out_link = AddMidiLink ( midi, MLTYPE_Sender,
  443.                 MLINK_Location, "out.0",
  444.                 MLINK_Comment, "Drum Output",
  445.                 MLINK_Name, "Out",
  446.                 TAG_END ))
  447.     {   ErrorF( window, "Unable to create MIDI link." );
  448.         LEAVE;
  449.     }
  450.  
  451.     unless (drum_player = CreatePlayer(
  452.                 PLAYER_Name, APPNAME,
  453.                 PLAYER_Conductor, "Main",
  454.                 PLAYER_Hook, &myHook,
  455.                 PLAYER_Priority, 0,
  456.                 TAG_DONE ))
  457.     {   ErrorF( window, "Unable to create RealTime player." );
  458.         LEAVE;
  459.     }
  460.  
  461.         /* open the window */
  462.  
  463.     window = OpenWindowTags(
  464.                 NULL,
  465.                 WA_Width,   640,
  466.                 WA_InnerHeight, 178,
  467.                 WA_IDCMP,   IDCMP_CLOSEWINDOW | IDCMP_MOUSEBUTTONS |
  468.                             IDCMP_MOUSEMOVE | IDCMP_MENUPICK |
  469.                             IDCMP_GADGETUP | IDCMP_GADGETDOWN |
  470.                             IDCMP_RAWKEY | IDCMP_ACTIVEWINDOW |
  471.                             IDCMP_INACTIVEWINDOW | IDCMP_IDCMPUPDATE |
  472.                             SLIDERIDCMP | BUTTONIDCMP, /* help ? */
  473.                 WA_Flags,   WFLG_DRAGBAR | WFLG_DEPTHGADGET |
  474.                             WFLG_CLOSEGADGET | WFLG_REPORTMOUSE |
  475.                             WFLG_NOCAREREFRESH | WFLG_ACTIVATE |
  476.                             WFLG_SMART_REFRESH | WFLG_NEWLOOKMENUS,
  477.                 WA_Title,   "Drum Machine",
  478.                 WA_ScreenTitle, "©1992 Sylvan Technical Arts" );
  479.  
  480.     unless (window)
  481.     {   ErrorF( window, "Unable to open window." );
  482.         LEAVE;
  483.     }
  484.  
  485.     unless (vi = GetVisualInfo( window->WScreen, TAG_DONE))
  486.     {   ErrorF( window, "Unable to get screen information." );
  487.         LEAVE;
  488.     }
  489.  
  490.     unless (dri = GetScreenDrawInfo( window->WScreen ))
  491.     {   ErrorF( window, "Unable to get screen information." );
  492.         LEAVE;
  493.     }
  494.  
  495.     rp = window->RPort;
  496.     dri_pens = dri->dri_Pens;
  497.  
  498.         /* create off-screen bitmap */
  499.  
  500.     unless ( off_bm = AllocBitMap( 32, window->Height, rp->BitMap->Depth, 0, NULL ))
  501.     {   ErrorF( window, "Unable to create off-screen bitmap." );
  502.         LEAVE;
  503.     }
  504.  
  505.     InitRastPort( &off_rp );
  506.     off_rp.BitMap = off_bm;
  507.     off_shine_rp = off_rp; SetAPen( &off_shine_rp, dri_pens[ SHINEPEN ] );
  508.     off_shade_rp = off_rp; SetAPen( &off_shade_rp, dri_pens[ SHADOWPEN ] );
  509.  
  510.         /* now, create the gadgets */
  511.  
  512.     unless (last_gadget = context_gadget = CreateContext( &first_gadget ))
  513.     {   ErrorF( window, "Unable to create gadgets." );
  514.         LEAVE;
  515.     }
  516.  
  517.     ng.ng_VisualInfo = vi;
  518.     ng.ng_TextAttr = &topaz8;
  519.  
  520.     for (row = 0; row < DRUM_ROWS; row++)
  521.     {
  522.             /* initialize the drum kit while we're at it */
  523.  
  524.         current_pattern->dpSetup[ row ].name[0] = 0;
  525.         current_pattern->dpSetup[ row ].key = 40 + row;
  526.         current_pattern->dpSetup[ row ].flags = 0;
  527.  
  528.             /* general parameters for each row */
  529.  
  530.         ng.ng_TopEdge = window->BorderTop + 12 + row * ROW_HEIGHT;
  531.         ng.ng_Height = ROW_HEIGHT - 1;
  532.         ng.ng_UserData = (APTR)row;             /* user data is row #           */
  533.  
  534.             /* Mute Button */
  535.  
  536.         ng.ng_LeftEdge = 7;
  537.         ng.ng_Width = 16;
  538.         ng.ng_GadgetText = "M";
  539.         ng.ng_Flags = PLACETEXT_IN;
  540.         ng.ng_GadgetID = ID_MUTE;
  541.         unless (last_gadget = CreateGadget( BUTTON_KIND, last_gadget, &ng,
  542.                 GA_Immediate, TRUE,
  543.                 TAG_DONE ))
  544.         {   ErrorF( window, "Unable to create gadgets." );
  545.             LEAVE;
  546.         }
  547.  
  548.             /* solo button */
  549.  
  550.         ng.ng_LeftEdge = 24;
  551.         ng.ng_GadgetText = "S";
  552.         ng.ng_GadgetID = ID_SOLO;
  553.         unless (last_gadget = CreateGadget( BUTTON_KIND, last_gadget, &ng,
  554.                 GA_Immediate, TRUE,
  555.                 TAG_DONE ))
  556.         {   ErrorF( window, "Unable to create gadgets." );
  557.             LEAVE;
  558.         }
  559.  
  560.             /* preview button */
  561.  
  562.         ng.ng_LeftEdge = 198;
  563.         ng.ng_GadgetText = "*";
  564.         ng.ng_GadgetID = ID_HIT;
  565.         unless (last_gadget = CreateGadget( BUTTON_KIND, last_gadget, &ng,
  566.                 GA_Immediate, TRUE,
  567.                 TAG_DONE ))
  568.         {   ErrorF( window, "Unable to create gadgets." );
  569.             LEAVE;
  570.         }
  571.  
  572.             /* drum name */
  573.  
  574.         ng.ng_LeftEdge = 42;
  575.         ng.ng_Width = 90;
  576.         if (row == 0) ng.ng_GadgetText = "Name"; else ng.ng_GadgetText = NULL;
  577.         ng.ng_Flags = PLACETEXT_ABOVE;
  578.         ng.ng_GadgetID = ID_PNAME;
  579.         unless (last_gadget = CreateGadget( STRING_KIND, last_gadget, &ng,
  580.                 GTST_MaxChars, 24,
  581.                 TAG_DONE ))
  582.         {   ErrorF( window, "Unable to create gadgets." );
  583.             LEAVE;
  584.         }
  585.         name_gadgets[ row ] = last_gadget;
  586.  
  587.             /* key number */
  588.  
  589.         ng.ng_LeftEdge = 135;
  590.         ng.ng_Width = 38;
  591.         if (row == 0) ng.ng_GadgetText = "   Note #";
  592.         ng.ng_Flags = PLACETEXT_ABOVE;
  593.         ng.ng_GadgetID = ID_KEY;
  594.         unless (last_gadget = CreateGadget( INTEGER_KIND, last_gadget, &ng,
  595.                 GTIN_MaxChars, 3,
  596.                 GTIN_Number, current_pattern->dpSetup[ row ].key,
  597.                 TAG_DONE ))
  598.         {   ErrorF( window, "Unable to create gadgets." );
  599.             LEAVE;
  600.         }
  601.         key_gadgets[ row ] = last_gadget;
  602.  
  603.             /* key number increment */
  604.  
  605.         ng.ng_LeftEdge = 173;
  606.         ng.ng_Width = 22;
  607.         ng.ng_GadgetText = NULL;
  608.         ng.ng_GadgetID = ID_KEYINC + (row << 8);
  609.  
  610.         unless (arrow_gadget = make_arrow( &ng, arrow_gadget, 1, 127, current_pattern->dpSetup[ row ].key ))
  611.         {   ErrorF( window, "Unable to create gadgets." );
  612.             LEAVE;
  613.         }
  614.         keyinc_gadgets[ row ] = arrow_gadget;
  615.  
  616.         if (first_arrow == NULL) first_arrow = arrow_gadget;
  617.     }
  618.  
  619.         /* general parameters for bottom row */
  620.  
  621.     ng.ng_TopEdge = window->BorderTop + 22 + DRUM_ROWS * ROW_HEIGHT;
  622.     ng.ng_Height = 14;
  623.     ng.ng_UserData = NULL;
  624.  
  625. #if 0
  626.         /* File Name indicator */
  627.  
  628.     ng.ng_LeftEdge = 8;
  629.     ng.ng_Width = 147;
  630.     ng.ng_GadgetText = NULL;
  631.     ng.ng_Flags = PLACETEXT_IN;
  632.     ng.ng_GadgetID = ID_FILE;
  633.     unless (last_gadget = CreateGadget( TEXT_KIND, last_gadget, &ng,
  634.             GTTX_Text,      "« No File »",
  635.             GTTX_Border,    TRUE,
  636.             GTTX_Clipped,   TRUE,
  637.             TAG_DONE ))
  638.     {   ErrorF( window, "Unable to create gadgets." );
  639.         LEAVE;
  640.     }
  641. #endif
  642.  
  643.         /* Play button */
  644.  
  645.     ng.ng_LeftEdge = 7 /* 162 */;
  646.     ng.ng_Width = 31;
  647.     ng.ng_GadgetText = ">";
  648.     ng.ng_Flags = PLACETEXT_IN;
  649.     ng.ng_GadgetID = ID_PLAY;
  650.     unless (last_gadget = CreateGadget( BUTTON_KIND, last_gadget, &ng, TAG_DONE ))
  651.     {   ErrorF( window, "Unable to create gadgets." );
  652.         LEAVE;
  653.     }
  654.  
  655.         /* Stop button */
  656.  
  657.     ng.ng_LeftEdge = 38 /* 193 */;
  658.     ng.ng_Width = 31;
  659.     ng.ng_GadgetText = "[]";
  660.     ng.ng_Flags = PLACETEXT_IN;
  661.     ng.ng_GadgetID = ID_STOP;
  662.     unless (last_gadget = CreateGadget( BUTTON_KIND, last_gadget, &ng, TAG_DONE ))
  663.     {   ErrorF( window, "Unable to create gadgets." );
  664.         LEAVE;
  665.     }
  666.  
  667.         /* Step Fwd button */
  668.  
  669.     ng.ng_LeftEdge = 73 /* 228 */;
  670.     ng.ng_Width = 22;
  671.     ng.ng_GadgetText = "«";
  672.     ng.ng_Flags = PLACETEXT_IN;
  673.     ng.ng_GadgetID = ID_STEPBACK;
  674.     unless (last_gadget = CreateGadget( BUTTON_KIND, last_gadget, &ng,
  675.         GA_Immediate, TRUE,
  676.         TAG_DONE ))
  677.     {   ErrorF( window, "Unable to create gadgets." );
  678.         LEAVE;
  679.     }
  680.  
  681.         /* Stop button */
  682.  
  683.     ng.ng_LeftEdge = 95 /* 250 */;
  684.     ng.ng_Width = 22;
  685.     ng.ng_GadgetText = "»";
  686.     ng.ng_Flags = PLACETEXT_IN;
  687.     ng.ng_GadgetID = ID_STEPFWD;
  688.     unless (last_gadget = CreateGadget( BUTTON_KIND, last_gadget, &ng,
  689.         GA_Immediate, TRUE,
  690.         TAG_DONE ))
  691.     {   ErrorF( window, "Unable to create gadgets." );
  692.         LEAVE;
  693.     }
  694.  
  695.         /* Tempo Slider */
  696.  
  697.     ng.ng_LeftEdge = 197;
  698.     ng.ng_Width = 95;
  699.     ng.ng_GadgetText = "Tempo:   ";
  700.     ng.ng_Flags = PLACETEXT_LEFT;
  701.     ng.ng_GadgetID = ID_TEMPO;
  702.     unless (last_gadget = CreateGadget( SLIDER_KIND, last_gadget, &ng,
  703.                 GTSL_Min,           10,
  704.                 GTSL_Max,           300,
  705.                 GTSL_Level,         current_tempo,
  706.                 GTSL_LevelFormat,   "%3.3ld",
  707.                 GTSL_MaxLevelLen,   3,
  708.                 GA_Immediate,       TRUE,
  709.                 GA_RelVerify,       TRUE,
  710.                 GA_FollowMouse,     TRUE,
  711.                 TAG_DONE ))
  712.     {   ErrorF( window, "Unable to create gadgets." );
  713.         LEAVE;
  714.     }
  715.  
  716.         /* Channel Control */
  717.  
  718.     ng.ng_LeftEdge = 367 /* 370 */;
  719.     ng.ng_Width = 38 /* 64 */;
  720.     ng.ng_GadgetText = "Channel:";
  721.     ng.ng_Flags = PLACETEXT_LEFT;
  722.     ng.ng_GadgetID = ID_CHANNEL;
  723.     unless (last_gadget = CreateGadget( INTEGER_KIND, last_gadget, &ng,
  724.             GTIN_MaxChars, 2,
  725.             GTIN_Number, current_channel + 1,
  726.             TAG_DONE ))
  727.     {   ErrorF( window, "Unable to create gadgets." );
  728.         LEAVE;
  729.     }
  730.     channel_string = last_gadget;
  731.  
  732.         /* Channel increment */
  733.  
  734.     ng.ng_LeftEdge += ng.ng_Width;
  735.     ng.ng_Width = 22;
  736.     ng.ng_GadgetText = NULL;
  737.     ng.ng_GadgetID = ID_CHANNELINC;
  738.  
  739.     unless (arrow_gadget = make_arrow( &ng, arrow_gadget, 0, 15, current_channel ))
  740.     {   ErrorF( window, "Unable to create gadgets." );
  741.         LEAVE;
  742.     }
  743.     channel_arrow = arrow_gadget;
  744.  
  745.         /* Velocity Slider */
  746.  
  747.     ng.ng_LeftEdge = 541 /* 545 */;
  748.     ng.ng_Width = 88 /* 84 */;
  749.     ng.ng_GadgetText = "Velocity:   ";
  750.     ng.ng_Flags = PLACETEXT_LEFT;
  751.     ng.ng_GadgetID = ID_VELOCITY;
  752.     unless (last_gadget = CreateGadget( SLIDER_KIND, last_gadget, &ng,
  753.                 GTSL_Min,   1,
  754.                 GTSL_Max,   127,
  755.                 GTSL_Level, current_velocity,
  756.                 GTSL_MaxLevelLen,   3,
  757.                 GTSL_LevelFormat,   "%3.3ld",
  758.                 GA_RelVerify,       TRUE,
  759.                 TAG_DONE ))
  760.     {   ErrorF( window, "Unable to create gadgets." );
  761.         LEAVE;
  762.     }
  763.  
  764.     AddGList( window, first_arrow, -1, -1, NULL );
  765.     AddGList( window, first_gadget, -1, -1, NULL );
  766.  
  767.     gadgets_added = TRUE;
  768.     GT_RefreshWindow( window, NULL );
  769.     RefreshGList( first_arrow, window, NULL, -1 );
  770.  
  771.         /* render the non-gadget window contents */
  772.  
  773.     matrix_top = window->BorderTop + 2;
  774.     matrix_bottom = window->BorderTop + 18 + row * ROW_HEIGHT;
  775.     matrix_y_org = matrix_top + 16;
  776.     matrix_height = matrix_bottom - matrix_top;
  777.  
  778.     column_width = clamp( 8, MAX_MATRIX_WIDTH / total_columns, 31 );
  779.     matrix_width = column_width * total_columns;
  780.  
  781.     SetDrPt( rp, 0xaaaa );
  782.     for (row = 0; row < DRUM_ROWS; row++)
  783.     {   int             y = matrix_y_org + row * ROW_HEIGHT;
  784.  
  785.         SetAPen( rp, dri_pens[ SHADOWPEN ] );
  786.         Move( rp, 214, y );
  787.         Draw( rp, 635, y );
  788.  
  789.         SetAPen( rp, dri_pens[ SHINEPEN ] );
  790.         Move( rp, 214, y + 1 );
  791.         Draw( rp, 635, y + 1 );
  792.     }
  793.     SetDrPt( rp, (UWORD)-1 );
  794.  
  795.     for (column = 0; column <= total_columns; column++)
  796.     {   render_column( column, TRUE );
  797.     }
  798.     draw_titlebar_scale( );
  799.  
  800.     SetAPen( rp, dri_pens[ SHADOWPEN ] );
  801.     Move( rp, MATRIX_X, matrix_top - 1 );
  802.     Draw( rp, MATRIX_X + total_columns * column_width, matrix_top - 1 );
  803.  
  804.     SetAPen( rp, dri_pens[ SHADOWPEN ] );
  805.     Move( rp, window->BorderLeft, matrix_bottom );
  806.     Draw( rp, 635, matrix_bottom );
  807.  
  808.     SetAPen( rp, dri_pens[ SHINEPEN ] );
  809.     Move( rp, window->BorderLeft, matrix_bottom + 1 );
  810.     Draw( rp, 635, matrix_bottom + 1 );
  811.  
  812.         /* Create menus */
  813.  
  814.     unless (menu_strip = CreateMenus( menu_list, GTMN_FrontPen, 0, TAG_END )) LEAVE;
  815.     unless (LayoutMenus( menu_strip, vi, GTMN_NewLookMenus, TRUE, TAG_END ) ) LEAVE;
  816.     SetMenuStrip( window, menu_strip );
  817.  
  818.         /*  Create ASL File requester */
  819.  
  820.     unless (filereq = AllocAslRequestTags( ASL_FileRequest,
  821.             ASLFR_Window,       window,
  822.             ASLFR_SleepWindow,  TRUE,
  823.             ASLFR_RejectIcons,  TRUE,
  824.             TAG_DONE ))
  825.     {   ErrorF( window, "Unable to create file requester." );
  826.         LEAVE;
  827.     }
  828.  
  829.     while (running)
  830.     {   struct IntuiMessage msg_copy;
  831.         struct IntuiMessage *msg;
  832.         struct Gadget       *g;
  833.         struct TagItem      temp_tags[8],       /* hold tags after replymsg     */
  834.                             *tag,
  835.                             *taglist;
  836.         WORD                gadget_id,
  837.                             gadget_value;
  838.  
  839.         LONG signals;
  840.  
  841.         signals = Wait( (1 << window->UserPort->mp_SigBit)
  842.                         | SIGBREAKF_CTRL_F
  843.                         | SIGBREAKF_CTRL_E );
  844.  
  845.         if (signals & SIGBREAKF_CTRL_F)         /* clock signal                 */
  846.         {
  847.             column = playback_column;
  848.             playback_column = column_pos;
  849.  
  850.             play_drum_note( ¤t_pattern->dpGrid[ playback_column ], TRUE );
  851.             play_drum_note( ¤t_pattern->dpGrid[ playback_column ], FALSE );
  852.  
  853.             render_column( column, FALSE );
  854.             render_column( playback_column, TRUE );
  855.         }
  856.  
  857.         if (signals & SIGBREAKF_CTRL_E)         /* state change signal          */
  858.         {
  859.             clock_state = drum_player->pl_Source->cdt_State;
  860.  
  861.             switch (clock_state) {
  862.  
  863.             case CONDSTATE_STOPPED:
  864.  
  865.                 break;
  866.  
  867.             case CONDSTATE_PAUSED:
  868.  
  869.                 break;
  870.  
  871.             case CONDSTATE_LOCATE:
  872.  
  873.                 render_column( playback_column, FALSE );
  874.                 locate_to_tick ( drum_player->pl_Source->cdt_ClockTime );
  875.                 playback_column = column_pos;
  876.  
  877.                 SetPlayerAttrs( drum_player, PLAYER_Ready, TRUE, TAG_END );
  878.                 break;
  879.  
  880.             case CONDSTATE_RUNNING:
  881.  
  882.                 render_column( playback_column, FALSE );
  883.                 locate_to_tick ( drum_player->pl_Source->cdt_ClockTime );
  884.                 playback_column = column_pos;
  885.  
  886.                 SetPlayerAttrs( drum_player, PLAYER_Ready, TRUE, TAG_END );
  887.  
  888.                 Signal( main_task, SIGBREAKF_CTRL_F );
  889.                 break;
  890.             }
  891.         }
  892.  
  893.         while (msg = GT_GetIMsg( window->UserPort ))
  894.         {   msg_copy = *msg;
  895.  
  896.             if (msg_copy.Class == IDCMPUPDATE)
  897.                 CopyMem( msg->IAddress, temp_tags, sizeof temp_tags );
  898.  
  899.             GT_ReplyIMsg( msg );
  900.  
  901.             switch ( msg_copy.Class ) {
  902.             case IDCMP_CLOSEWINDOW: LEAVE;
  903.  
  904.             case IDCMP_ACTIVEWINDOW:
  905.             case IDCMP_INACTIVEWINDOW:
  906.                 draw_titlebar_scale( );
  907.                 break;
  908.  
  909.             case IDCMP_MENUPICK:
  910.                 if (drag_state)
  911.                 {   shadow_drag( drag_column, drag_anchor_column );
  912.                     drag_state = 0;
  913.                 }
  914.  
  915.                 while (msg_copy.Code != MENUNULL && msg_copy.IDCMPWindow)
  916.                 {   struct MenuItem *menuitem;
  917.  
  918.                     if (menuitem = ItemAddress( msg_copy.IDCMPWindow->MenuStrip,msg_copy.Code ))
  919.                     {   void (*userdata)( struct IntuiMessage * );
  920.  
  921.                         if (userdata = (void (*)( struct IntuiMessage *))GTMENUITEM_USERDATA(menuitem))
  922.                         {   (userdata)( &msg_copy );
  923.                         }
  924.                         msg_copy.Code = menuitem->NextSelect;
  925.                     }
  926.                     else break;
  927.                 }
  928.  
  929.                 break;
  930.  
  931.             case IDCMP_IDCMPUPDATE:
  932.                 taglist = temp_tags;
  933.  
  934.                 while (tag = NextTagItem( &taglist ))
  935.                 {
  936.                     switch (tag->ti_Tag) {
  937.                     case GA_ID: gadget_id = tag->ti_Data; break;
  938.                     }
  939.                 }
  940.  
  941.                 switch ( gadget_id & 0xff ) {
  942.                 case ID_KEYINC:
  943.  
  944.                     gadget_value = msg_copy.Code;
  945.                     row = gadget_id >> 8;
  946.                     current_pattern->dpSetup[ row ].key = gadget_value;
  947.                     GT_SetGadgetAttrs( key_gadgets[ row ], window, NULL,
  948.                             GTIN_Number, gadget_value,
  949.                             TAG_DONE );
  950.  
  951.                     if (last_qualifier & (IEQUALIFIER_LALT|IEQUALIFIER_RALT))
  952.                         preview_row( row );
  953.  
  954.                     break;
  955.  
  956.                 case ID_CHANNELINC:
  957.                     gadget_value = msg_copy.Code;
  958.                     current_channel = clamp( 0, gadget_value, 15);
  959.                     GT_SetGadgetAttrs( channel_string, window, NULL,
  960.                             GTIN_Number, current_channel + 1,
  961.                             TAG_DONE );
  962.                     break;
  963.  
  964.                 }
  965.                 break;
  966.  
  967.             case IDCMP_GADGETDOWN:
  968.                 g = (struct Gadget *)msg_copy.IAddress;
  969.                 row = (int)g->UserData;
  970.  
  971.                 switch (g->GadgetID) {
  972.                 case ID_HIT:        preview_row( row ); break;
  973.                 case ID_STEPFWD:    step_play( 1 ); break;
  974.                 case ID_STEPBACK:   step_play( -1 ); break;
  975.                 case ID_TEMPO:      tempo_drag = TRUE; break;
  976.  
  977.                 };
  978.                 break;
  979.  
  980.             case IDCMP_GADGETUP:
  981.                 g = (struct Gadget *)msg_copy.IAddress;
  982.                 row = (int)g->UserData;
  983.  
  984.                 switch (g->GadgetID) {
  985.  
  986.                 case ID_PLAY:
  987.                     SetConductorState( drum_player, CONDSTATE_LOCATE, 0 );
  988.                     break;
  989.  
  990.                 case ID_STOP:
  991.                     SetConductorState( drum_player, CONDSTATE_STOPPED, 0 );
  992.                     break;
  993.  
  994.                 case ID_KEY:
  995.                     current_pattern->dpSetup[ row ].key =
  996.                         ((struct StringInfo *)g->SpecialInfo)->LongInt;
  997.  
  998.                     SetGadgetAttrs( keyinc_gadgets[ row ], window, NULL,
  999.                         ARROW_Current, current_pattern->dpSetup[ row ].key,
  1000.                         TAG_END );
  1001.  
  1002.                     break;
  1003.  
  1004.                 case ID_CHANNEL:
  1005.  
  1006.                     current_channel =
  1007.                         clamp(0, ((struct StringInfo *)g->SpecialInfo)->LongInt - 1, 15);
  1008.  
  1009.                     GT_SetGadgetAttrs( key_gadgets[ row ], window, NULL,
  1010.                             GTIN_Number, current_channel + 1,
  1011.                             TAG_DONE );
  1012.  
  1013.                     SetGadgetAttrs( channel_arrow, window, NULL,
  1014.                         ARROW_Current, current_channel,
  1015.                         TAG_END );
  1016.  
  1017.                     break;
  1018.  
  1019.                 case ID_PNAME:
  1020.  
  1021.                     strncpy(current_pattern->dpSetup[ row ].name,
  1022.                             ((struct StringInfo *)g->SpecialInfo)->Buffer,
  1023.                             sizeof current_pattern->dpSetup[ row ].name );
  1024.                     break;
  1025.  
  1026.                 case ID_TEMPO:
  1027.                     current_tempo = msg_copy.Code;
  1028.                     tempo_drag = FALSE;
  1029.                     calc_tempo();
  1030.                     break;
  1031.  
  1032.                 case ID_VELOCITY:
  1033.                     current_velocity = msg_copy.Code;
  1034.                     break;
  1035.  
  1036.                 };
  1037.                 break;
  1038.  
  1039.             case IDCMP_MOUSEMOVE:
  1040.  
  1041.                 if (tempo_drag)
  1042.                 {   current_tempo = msg_copy.Code;
  1043.                     calc_tempo();
  1044.                 }
  1045.                 else if (drag_state)
  1046.                 {
  1047.                     column = msg_copy.MouseX - MATRIX_X + column_width / 2;
  1048.                     column /= column_width;
  1049.  
  1050.                     if (column >= 0 && column < total_columns && column != drag_column )
  1051.                     {   shadow_drag( drag_column, drag_anchor_column );
  1052.                         drag_column = column;
  1053.                         shadow_drag( drag_column, drag_anchor_column );
  1054.                     }
  1055.                 }
  1056.                 break;
  1057.  
  1058.             case IDCMP_MOUSEBUTTONS:
  1059.  
  1060.                 if (msg_copy.Code == SELECTDOWN)
  1061.                 {
  1062.                     row = msg_copy.MouseY - window->BorderTop - 12;
  1063.                     column = msg_copy.MouseX - MATRIX_X + column_width / 2;
  1064.  
  1065.                     if (row >= 0 && column >= 0)
  1066.                     {   row /= ROW_HEIGHT;
  1067.                         column /= column_width;
  1068.  
  1069.                         if (column == total_columns) column = 0;
  1070.  
  1071.                         if (row < DRUM_ROWS && column < total_columns)
  1072.                         {   struct DrumColumn   *dc = ¤t_pattern->dpGrid[ column ];
  1073.  
  1074.                             if (dc->note[ row ] > 0)
  1075.                             {   dc->note[ row ] = 0;
  1076.                                 drag_state = DRAG_RESET;
  1077.                             }
  1078.                             else
  1079.                             {   dc->note[ row ] = current_velocity;
  1080.                                 drag_state = DRAG_SET;
  1081.                             }
  1082.  
  1083.                             drag_column = drag_anchor_column = column;
  1084.                             drag_row = row;
  1085.  
  1086.                             render_column( column, TRUE );
  1087.                             if (column == 0) render_column( total_columns, FALSE );
  1088.  
  1089.                             if (dc->note[ row ] > 0 &&
  1090.                                 (msg_copy.Qualifier & (IEQUALIFIER_LALT|IEQUALIFIER_RALT)))
  1091.                                     preview_row( row );
  1092.                         }
  1093.                     }
  1094.                 }
  1095.                 else if (msg_copy.Code == SELECTUP)
  1096.                 {
  1097.                     if (drag_state && drag_column != drag_anchor_column)
  1098.                     {   int i;
  1099.  
  1100.                         shadow_drag( drag_column, drag_anchor_column );
  1101.  
  1102.                         if (drag_column < drag_anchor_column)
  1103.                         {   i = drag_column;
  1104.                             drag_column = drag_anchor_column;
  1105.                             drag_anchor_column = i;
  1106.                         }
  1107.  
  1108.                         for (i=drag_anchor_column; i<=drag_column; i++)
  1109.                         {   struct DrumColumn   *dc;
  1110.  
  1111.                             column = (i == total_columns) ? 0 : i;
  1112.  
  1113.                             dc = ¤t_pattern->dpGrid[ column ];
  1114.  
  1115.                             if (drag_state == DRAG_RESET) dc->note[ row ] = 0;
  1116.                             else dc->note[ row ] = current_velocity;
  1117.  
  1118.                             render_column( column, TRUE );
  1119.                             if (column == 0) render_column( total_columns, FALSE );
  1120.                         }
  1121.                     }
  1122.                     drag_state = 0;
  1123.                 }
  1124.                 break;
  1125.  
  1126.             case IDCMP_RAWKEY:
  1127.  
  1128.                 if (drag_state)
  1129.                 {   shadow_drag( drag_column, drag_anchor_column );
  1130.                     drag_state = 0;
  1131.                 }
  1132.  
  1133.                 last_qualifier = msg_copy.Qualifier;
  1134.  
  1135.                 switch (msg_copy.Code) {
  1136.                 case 0x4e:                      /* cursor forward               */
  1137.                     step_play( 1 );
  1138.                     break;
  1139.  
  1140.                 case 0x4f:                      /* cursor backwards             */
  1141.                     step_play( -1 );
  1142.                     break;
  1143.  
  1144.                 case 1:                         /* number keys are drums...     */
  1145.                 case 2:
  1146.                 case 3:
  1147.                 case 4:
  1148.                 case 5:
  1149.                 case 6:
  1150.                 case 7:
  1151.                 case 8:
  1152.                 case 9:
  1153.                 case 10:
  1154.                     preview_row( msg_copy.Code - 1 );
  1155.                     /* Holding control key enters or deleted from score at current
  1156.                         playback position, minus a bit... */
  1157.                     break;
  1158.                 };
  1159.             };
  1160.         }
  1161.     }
  1162. exitit:
  1163.     if (drum_player) DeletePlayer( drum_player );
  1164.     if (midi) DeleteMidi( midi );
  1165.  
  1166.     FreeBitMap( off_bm );
  1167.  
  1168.     if (filereq) FreeAslRequest( filereq );
  1169.  
  1170.     if (window) { ClearMenuStrip( window ); CloseWindow( window ); }
  1171.     if (menu_strip) FreeMenus( menu_strip );
  1172.  
  1173. /*  if (gadgets_added) RemoveGList( window, first_gadget, -1 ); */
  1174.     FreeGadgets( first_gadget );
  1175.     DisposeObjectList( (struct List *)&arrow_list );
  1176.     FreeScreenDrawInfo( window->WScreen, dri );
  1177.     FreeVisualInfo( vi );
  1178.  
  1179.     freeArrowButtonClass(aclass);               /* delete boopsi classes        */
  1180.  
  1181.     CloseLibrary( IFFParseBase );
  1182.     CloseLibrary( RealTimeBase );
  1183.     CloseLibrary( CamdBase );
  1184.     CloseLibrary( GfxBase );
  1185.     CloseLibrary( UtilityBase );
  1186.     CloseLibrary( AslBase );
  1187.     CloseLibrary( GadToolsBase );
  1188.     CloseLibrary( IntuitionBase );
  1189. }
  1190.  
  1191. /* ============================================================================ *
  1192.                                    Editing Routines
  1193.  * ============================================================================ */
  1194.  
  1195.     /* add an object to a list */
  1196.  
  1197. ULONG AddTailObject( Object *o, struct List *list)
  1198. {   return DoMethod( o, OM_ADDTAIL, list );
  1199. }
  1200.  
  1201. void DisposeObjectList( struct List *l )
  1202. {   APTR                *obj_state = (APTR)l->lh_Head,
  1203.                         *obj;
  1204.  
  1205.     while ( obj = NextObject( &obj_state ) )    /* iterate through list         */
  1206.     {   DoMethod( (Object *)obj, OM_REMOVE );   /* remove from list             */
  1207.         DisposeObject( obj );
  1208.     }
  1209. }
  1210.  
  1211. struct TagItem arrow_map[] = {
  1212.     ARROW_Current,  ICSPECIAL_CODE,
  1213.     TAG_END,
  1214. };
  1215.  
  1216. struct Gadget *make_arrow( struct NewGadget *ng, struct Gadget *prev, int min, int max, int current )
  1217. {   struct Gadget       *g;
  1218.  
  1219.     g = (struct Gadget *)NewObject(aclass,NULL,
  1220.             GA_Left,        ng->ng_LeftEdge,
  1221.             GA_Top,         ng->ng_TopEdge,
  1222.             GA_Width,       ng->ng_Width,
  1223.             GA_Height,      ng->ng_Height,
  1224.             GA_DrawInfo,    dri,
  1225.             GA_ID,          ng->ng_GadgetID,
  1226.             GA_UserData,    ng->ng_UserData,
  1227.  
  1228.             ARROW_Min,      min,
  1229.             ARROW_Max,      max,
  1230.             ARROW_Current,  current,
  1231.             ARROW_IncreaseRate, 5,
  1232.             ARROW_MaxRate,  20,
  1233.  
  1234.             ICA_TARGET,     ICTARGET_IDCMP,
  1235.             ICA_MAP,        arrow_map,
  1236.  
  1237.             prev ? GA_Previous : TAG_IGNORE, prev,
  1238.             TAG_END );
  1239.  
  1240.     if (g) AddTailObject( (Object *)g, (struct List *)&arrow_list );
  1241.  
  1242.     return g;
  1243. }
  1244.  
  1245. void draw_titlebar_scale( void )
  1246. {   int                 column;
  1247.     int                 y = window->BorderTop - 2;
  1248.  
  1249.     for (column = 0; column <= total_columns; column+= 2)
  1250.     {   int             x = MATRIX_X + column * column_width;
  1251.         int             h;
  1252.  
  1253.         if ((column & 0x07) == 0) h = 7;
  1254.         else if ((column & 0x03) == 0) h = 3;
  1255.         else h = 1;
  1256.  
  1257.         SetAPen( rp, dri_pens[ SHADOWPEN ] );
  1258.         Move( rp, x, y - h );
  1259.         Draw( rp, x, y );
  1260.  
  1261.         SetAPen( rp, dri_pens[ SHINEPEN ] );
  1262.         Move( rp, x + 1, y - h );
  1263.         Draw( rp, x + 1, y );
  1264.     }
  1265. }
  1266.  
  1267. void render_column( int column, int highlight )
  1268. {   int                 side_width = column_width / 2;
  1269.     int                 x = MATRIX_X + column * column_width;
  1270.     int                 row;
  1271.     struct DrumColumn   *dc = ¤t_pattern->dpGrid[ column >= total_columns ? 0 : column ];
  1272.  
  1273.     if (column != playback_column) highlight = FALSE;
  1274.  
  1275.     SetAPen( &off_rp, 0 );
  1276.     RectFill( &off_rp, 0, 0, column_width, matrix_height );
  1277.     SetAPen( &off_rp, dri_pens[ highlight ? HIGHLIGHTTEXTPEN : TEXTPEN ] );
  1278.  
  1279.     SetDrPt( &off_shine_rp, (x & 1) ? 0xaaaa : 0x5555 );
  1280.     SetDrPt( &off_shade_rp, (x & 1) ? 0xaaaa : 0x5555 );
  1281.  
  1282.         /* draw the individual rows */
  1283.  
  1284.     for (row = 0; row < DRUM_ROWS; row++)
  1285.     {   int             y = 16 + row * ROW_HEIGHT;
  1286.  
  1287.         Move( &off_shade_rp, 0, y );
  1288.         Draw( &off_shade_rp, column_width, y );
  1289.  
  1290.         Move( &off_shine_rp, 0, y + 1 );
  1291.         Draw( &off_shine_rp, column_width, y + 1 );
  1292.     }
  1293.  
  1294.     SetDrPt( &off_shine_rp, (UWORD)-1 );
  1295.     SetDrPt( &off_shade_rp, (UWORD)-1 );
  1296.  
  1297.     if (highlight)
  1298.     {   Move( &off_shine_rp, side_width - 1, 0 );
  1299.         Draw( &off_shine_rp, side_width - 1, matrix_height );
  1300.     }
  1301.     else
  1302.     {   Move( &off_shade_rp, side_width - 1, 0 );
  1303.         Draw( &off_shade_rp, side_width - 1, matrix_height );
  1304.     }
  1305.     Move( &off_shine_rp, side_width, 0 );
  1306.     Draw( &off_shine_rp, side_width, matrix_height );
  1307.  
  1308.     for (row = 0; row < DRUM_ROWS; row++)
  1309.     {   int             y = 16 + row * ROW_HEIGHT;
  1310.         int             hit,
  1311.                         dotsize;
  1312.  
  1313.         hit = dc->note[ row ];
  1314.         if (hit > 0)
  1315.         {   if (hit > 90) dotsize = 4;
  1316.             else if (hit > 60) dotsize = 3;
  1317.             else if (hit > 30) dotsize = 2;
  1318.             else dotsize = 1;
  1319.  
  1320.             RectFill( &off_rp,  side_width - dotsize - 1,
  1321.                                 y - dotsize,
  1322.                                 side_width + dotsize - 1,
  1323.                                 y + dotsize );
  1324.         }
  1325.     }
  1326.  
  1327.     BltBitMapRastPort(  off_bm, 0, 0,
  1328.                         rp, x - side_width + 1, matrix_top,
  1329.                         side_width + side_width, matrix_height,
  1330.                         0xC0 );
  1331. }
  1332.  
  1333. void render_all( void )
  1334. {   int                 column;
  1335.  
  1336.     for (column = 0; column <= total_columns; column++)
  1337.     {   render_column( column, TRUE );
  1338.     }
  1339.  
  1340.     /* also render gadgets */
  1341. }
  1342.  
  1343. void set_kit_gadgets( void )
  1344. {   int                 row;
  1345.  
  1346.     for (row = 0; row < DRUM_ROWS; row++)
  1347.     {
  1348.         GT_SetGadgetAttrs( key_gadgets[ row ], window, NULL,
  1349.                 GTIN_Number, current_pattern->dpSetup[ row ].key,
  1350.                 TAG_DONE );
  1351.  
  1352.         SetGadgetAttrs( keyinc_gadgets[ row ], window, NULL,
  1353.             ARROW_Current, current_pattern->dpSetup[ row ].key,
  1354.             TAG_END );
  1355.  
  1356.         GT_SetGadgetAttrs( name_gadgets[ row ], window, NULL,
  1357.                 GTST_String, current_pattern->dpSetup[ row ].name,
  1358.                 TAG_DONE );
  1359.     }
  1360. }
  1361.  
  1362. void preview_row( int row )
  1363. {   struct DrumColumn   preview_col;
  1364.  
  1365.     memset( &preview_col, 0, sizeof preview_col );
  1366.     preview_col.note[ row ] = current_velocity;
  1367.  
  1368.     play_drum_note( &preview_col, TRUE );
  1369.     play_drum_note( &preview_col, FALSE );
  1370. }
  1371.  
  1372. void play_drum_note( struct DrumColumn *dc, BOOL note_on )
  1373. {   int                 row;
  1374.     MidiMsg             mm;
  1375.  
  1376.     mm.mm_Status = MS_NoteOn | current_channel;
  1377.  
  1378.     for (row = 0; row < DRUM_ROWS; row++)
  1379.     {   if (dc->note[ row ] > 0 && !(current_pattern->dpSetup[ row ].flags & DRUMF_MUTE))
  1380.         {
  1381.             mm.mm_Data1 = current_pattern->dpSetup[ row ].key;
  1382.             mm.mm_Data2 = note_on ? dc->note[ row ] : 0;
  1383.  
  1384.             PutMidiMsg ( out_link, &mm );
  1385.         }
  1386.     }
  1387. }
  1388.  
  1389. void step_play( int dir )
  1390. {   int column = playback_column;
  1391.  
  1392.     if (dir > 0)
  1393.     {   playback_column++;
  1394.         if (playback_column >= total_columns) playback_column = 0;
  1395.     }
  1396.     else
  1397.     {   playback_column--;
  1398.         if (playback_column < 0) playback_column = total_columns - 1;
  1399.     }
  1400.  
  1401.     play_drum_note( ¤t_pattern->dpGrid[ playback_column ], TRUE );
  1402.     play_drum_note( ¤t_pattern->dpGrid[ playback_column ], FALSE );
  1403.  
  1404.     render_column( column, FALSE );
  1405.     render_column( playback_column, TRUE );
  1406. }
  1407.  
  1408. void shadow_drag( int col1, int col2 )
  1409. {   int                 x1 = MATRIX_X + col1 * column_width,
  1410.                         x2 = MATRIX_X + col2 * column_width;
  1411.     int                 y = matrix_y_org + drag_row * ROW_HEIGHT;
  1412.  
  1413.     if (drag_column != drag_anchor_column)
  1414.     {   SetDrMd( rp, COMPLEMENT );
  1415.         Move( rp, x1, y );
  1416.         Draw( rp, x2, y );
  1417.         SetDrMd( rp, JAM2 );
  1418.     }
  1419. }
  1420.  
  1421. /* ============================================================================ *
  1422.                                  Save / Load routines
  1423.  * ============================================================================ */
  1424.  
  1425.     /* REM: Is the list of sounds local or global?
  1426.         Samples should be global,
  1427.         sounds should be local...
  1428.     */
  1429.  
  1430.     /*  Print error message from IFFParse */
  1431.  
  1432. void IFFError(char *filename, int writing, int error)
  1433. {   char                *msg;
  1434.  
  1435.     switch ( error ) {
  1436.     case -100:                  msg = "Can't open file."; break;
  1437.     case IFFERR_EOF:            return;
  1438.     case IFFERR_EOC:            msg = "File is Incomplete";
  1439.     case IFFERR_NOSCOPE:        return;
  1440.  
  1441.     case -101:
  1442.     case IFFERR_NOMEM:          msg = "Out of Memory"; break;
  1443.  
  1444.     case IFFERR_READ:           msg = "Read Error"; break;
  1445.     case IFFERR_WRITE:          msg = "Write Error"; break;
  1446.     case IFFERR_SEEK:           msg = "Read Error"; break;
  1447.     case IFFERR_MANGLED:        msg = "File is Corrupt"; break;
  1448.     case IFFERR_SYNTAX:         msg = "File is Incomplete"; break;
  1449.     case IFFERR_NOTIFF:         msg = "Not an IFF File"; break;
  1450.  
  1451.     case IFFERR_NOHOOK:
  1452.     case IFF_RETURN2CLIENT:
  1453.     default:                    msg = "Internal Error"; break;
  1454.     }
  1455.  
  1456.     if (filename)
  1457.     {   ErrorF(
  1458.             window,
  1459.             writing ? "Error Writing File '%s':\n%s." : "Error Reading File '%s':\n%s.",
  1460.             filename,
  1461.             msg );
  1462.     }
  1463.     else
  1464.     {   ErrorF(
  1465.             window,
  1466.             writing ? "Error Writing to Clipboard:\n%s." : "Error Reading from Clipboard:\n%s",
  1467.             msg );
  1468.     }
  1469.  
  1470. }
  1471.  
  1472. WORD read_kit( struct IFFHandle *iff, struct DrumPattern *dp )
  1473. {   char                *tbuf;
  1474.     struct ContextNode  *cn;
  1475.     WORD                error = FALSE;
  1476.  
  1477.     unless (cn = CurrentChunk (iff)) return -102;       /* internal error */
  1478.     if (tbuf = AllocMem( cn->cn_Size, 0L ))
  1479.     {   if ((error = ReadChunkBytes( iff, tbuf, cn->cn_Size )) >= 0)
  1480.         {
  1481.             CopyMem( tbuf, &ReadKit, MIN( cn->cn_Size, sizeof ReadKit ) );
  1482.             CopyMem( &ReadKit, dp->dpSetup, sizeof dp->dpSetup ); /* temporary */
  1483.             error = 0;
  1484.         }
  1485.         FreeMem( tbuf, cn->cn_Size );
  1486.     }
  1487.     return error;
  1488. }
  1489.  
  1490. WORD read_pattern( struct IFFHandle *iff, struct DrumPattern *dp )
  1491. {   char                *tbuf;
  1492.     struct ContextNode  *cn;
  1493.     WORD                error = FALSE;
  1494.  
  1495.     unless (cn = CurrentChunk (iff)) return -102;       /* internal error */
  1496.     if (tbuf = AllocMem( cn->cn_Size, 0L ))
  1497.     {   if ((error = ReadChunkBytes( iff, tbuf, cn->cn_Size )) >= 0)
  1498.         {   int         i,j;
  1499.             struct DrumPattern  *pat;
  1500.             char        *array;
  1501.             WORD        array_bytes;
  1502.  
  1503.             pat = (struct DrumPattern *)tbuf;
  1504.         /*  dp->dpNumber = pat->dp_Number;  */  /* wrong? */
  1505.             dp->dpColumns = MIN( pat->dpColumns, MAX_COLUMNS );
  1506.             dp->dpRows = DRUM_ROWS;
  1507.  
  1508.             array = tbuf + offsetof (struct DrumPattern, dpSetup);
  1509.             array_bytes = cn->cn_Size - offsetof( struct DrumPattern, dpSetup );
  1510.  
  1511.             memset( &dp->dpGrid, 0, sizeof dp->dpGrid );
  1512.  
  1513.             for ( i=0; i<dp->dpColumns; i++)
  1514.             {
  1515.                 for (j=0; j<dp->dpRows; j++)
  1516.                 {   WORD    val = 0;
  1517.  
  1518.                     if (i < pat->dpColumns && j < pat->dpRows)
  1519.                     {   val = array[j];
  1520.                     }
  1521.                     dp->dpGrid[i].note[j] = val;
  1522.                 }
  1523.                 array += dp->dpRows;
  1524.             }
  1525.             error = 0;
  1526.         }
  1527.         FreeMem( tbuf, cn->cn_Size );
  1528.     }
  1529.     return error;
  1530. }
  1531.  
  1532. WORD write_kit( struct IFFHandle *iff, struct DrumPattern *dp )
  1533. {   int                 size = sizeof dp->dpSetup;
  1534.     WORD                error;
  1535.  
  1536.     if (error = PushChunk( iff, 0L, 'DKIT', size )) return error;
  1537.     if ((error = WriteChunkBytes (iff, (APTR) &dp->dpSetup, size)) != size) return error;
  1538.     if (error = PopChunk (iff)) return error;
  1539.     return 0;
  1540. }
  1541.  
  1542. WORD write_pattern( struct IFFHandle *iff, struct DrumPattern *dp )
  1543. {   int                 size1 = offsetof( struct DrumPattern, dpSetup),
  1544.                         size2 = sizeof (struct DrumColumn) * dp->dpColumns;
  1545.     WORD                error;
  1546.  
  1547.     if (error = PushChunk( iff, 0L, 'PTRN', size1 + size2 )) return error;
  1548.     if ((error = WriteChunkBytes( iff, (APTR) dp, size1 )) < 0) return error;
  1549.     if ((error = WriteChunkBytes( iff, (APTR) &dp->dpGrid, size2 )) < 0) return error;
  1550.     if (error = PopChunk (iff)) return error;
  1551.     return 0;
  1552. }
  1553.  
  1554. #if 0
  1555. BOOL read_song( void )
  1556. {   ;
  1557. }
  1558.  
  1559. BOOL write_song( void )
  1560. {   ;
  1561. }
  1562. #endif
  1563.  
  1564. static LONG         cmus_chunks[] = {
  1565.     'DRUM', 'DKIT',
  1566.     'DRUM', 'PTRN',
  1567. };
  1568.  
  1569. BOOL load_file( BOOL all )
  1570. {   struct IFFHandle    *iff = NULL;
  1571.     BOOL                error = 0;
  1572.     BPTR                fh = NULL,
  1573.                         olddir,
  1574.                         dlock = NULL;
  1575.  
  1576.         /*  Bring up the file requester */
  1577.  
  1578.     if (!AslRequestTags( (APTR)filereq,
  1579.                     ASLFR_TitleText,    all ? "Load Song" : "Load Pattern",
  1580.                     ASLFR_PositiveText, "Load",
  1581.                     TAG_DONE ))
  1582.     {   return 0;
  1583.     }
  1584.  
  1585.         /*  Lock the directory they typed */
  1586.  
  1587.     unless (dlock = Lock( filereq->rf_Dir, ACCESS_READ ))
  1588.     {   ErrorF( window, "Can't find directory: '%s'.", filereq->rf_Dir );
  1589.     }
  1590.  
  1591.         /*  CD to that directory and open the file */
  1592.  
  1593.     olddir = CurrentDir( dlock );
  1594.     fh = Open( filereq->rf_File, MODE_OLDFILE );
  1595.     CurrentDir( olddir );
  1596.  
  1597.         /*  Init IFF Stream */
  1598.  
  1599.     unless (fh) { error = -100; LEAVE; }
  1600.     unless (iff = AllocIFF()) { error = -101; LEAVE; }
  1601.     iff->iff_Stream = fh;
  1602.     InitIFFasDOS (iff);
  1603.  
  1604.         /*  Read the pattern */
  1605.  
  1606.     if (error = OpenIFF (iff, IFFF_WRITE)) LEAVE;
  1607.     if (error = StopChunks (iff, cmus_chunks, elementsof (cmus_chunks) / 2)) LEAVE;
  1608.  
  1609.     for (;;)
  1610.     {   struct ContextNode  *cn;
  1611.  
  1612.         if (error = ParseIFF (iff, IFFPARSE_SCAN))
  1613.         {   if (error == IFFERR_EOF) break;
  1614.             LEAVE;
  1615.         }
  1616.  
  1617.         unless (cn = CurrentChunk (iff))    /* shouldn't happen             */
  1618.         {   error = IFFERR_EOC;
  1619.             LEAVE;
  1620.         }
  1621.  
  1622.         switch (cn->cn_ID) {
  1623.         case 'DKIT':
  1624.             if (error = read_kit( iff, current_pattern )) LEAVE;
  1625.             break;
  1626.  
  1627.         case 'PTRN':
  1628.             if (error = read_pattern( iff, current_pattern )) LEAVE;
  1629.             break;
  1630.         }
  1631.     }
  1632.  
  1633. exitit:
  1634.  
  1635.         /*  Clean up */
  1636.  
  1637.     if (iff)
  1638.     {   CloseIFF (iff);
  1639.         FreeIFF (iff);
  1640.     }
  1641.     if (fh) Close( fh );
  1642.     if (dlock) UnLock( dlock );
  1643.  
  1644.         /*  Report Error */
  1645.  
  1646.     if (error)
  1647.     {   IFFError( filereq->rf_File, FALSE, error);
  1648.     }
  1649.  
  1650.     render_all();
  1651.     set_kit_gadgets();
  1652.  
  1653.     return error;
  1654. }
  1655.  
  1656. BOOL save_file( BOOL all )
  1657. {   struct IFFHandle    *iff = NULL;
  1658.     BOOL                error = 0;
  1659.     BPTR                fh = NULL,
  1660.                         olddir,
  1661.                         dlock = NULL;
  1662.  
  1663.         /*  Bring up the file requester */
  1664.  
  1665.     if (!AslRequestTags( (APTR)filereq,
  1666.                     ASLFR_TitleText,    all ? "Save Song" : "Save Pattern",
  1667.                     ASLFR_PositiveText, "Save",
  1668.                     ASLFR_DoSaveMode,   TRUE,
  1669.                     TAG_DONE ))
  1670.     {   return 0;
  1671.     }
  1672.  
  1673.         /*  Lock the directory they typed */
  1674.  
  1675.     unless (dlock = Lock( filereq->rf_Dir, ACCESS_READ ))
  1676.     {   ErrorF( window, "Can't find directory: '%s'.", filereq->rf_Dir );
  1677.     }
  1678.  
  1679.         /*  CD to that directory and open the file */
  1680.  
  1681.     olddir = CurrentDir( dlock );
  1682.     fh = Open( filereq->rf_File, MODE_NEWFILE );
  1683.     CurrentDir( olddir );
  1684.  
  1685.         /*  Init IFF Stream */
  1686.  
  1687.     unless (fh) { error = -100; LEAVE; }
  1688.     unless (iff = AllocIFF()) { error = -101; LEAVE; }
  1689.     iff->iff_Stream = fh;
  1690.     InitIFFasDOS (iff);
  1691.  
  1692.         /*  Write the pattern */
  1693.  
  1694.     if (error = OpenIFF (iff, IFFF_WRITE)) LEAVE;
  1695.     if (error = PushChunk (iff, 'DRUM', 'FORM', IFFSIZE_UNKNOWN)) LEAVE;
  1696.     if (error = write_kit( iff, current_pattern )) LEAVE;
  1697.     if (error = write_pattern( iff, current_pattern )) LEAVE;
  1698.     if (error = PopChunk (iff)) LEAVE;
  1699.  
  1700. exitit:
  1701.  
  1702.         /*  Clean up */
  1703.  
  1704.     if (iff)
  1705.     {   CloseIFF (iff);
  1706.         FreeIFF (iff);
  1707.     }
  1708.     if (fh) Close( fh );
  1709.     if (dlock) UnLock( dlock );
  1710.  
  1711.         /*  Report Error */
  1712.  
  1713.     if (error)
  1714.     {   IFFError( filereq->rf_File, TRUE, error);
  1715.     }
  1716.     return error;
  1717. }
  1718.  
  1719. /* ============================================================================ *
  1720.                                     Menu Functions
  1721.  * ============================================================================ */
  1722.  
  1723. void do_loadpattern( struct IntuiMessage *imsg )
  1724. {
  1725.     load_file( FALSE );
  1726. }
  1727.  
  1728. void do_savepattern( struct IntuiMessage *imsg )
  1729. {
  1730.     save_file( FALSE );
  1731. }
  1732.  
  1733. void do_about( struct IntuiMessage *imsg )
  1734. {   struct EasyStruct   es;
  1735.     APTR                dummyarg;
  1736.  
  1737.     es.es_StructSize = sizeof(struct EasyStruct);
  1738.     es.es_Flags = 0;
  1739.     es.es_Title = APPNAME " Program Information";
  1740.     es.es_TextFormat = "Drum Machine CAMD/RealTime Example\nBy Talin\n©1992 Sylvan Technical Arts";
  1741.     es.es_GadgetFormat = "Continue";
  1742.  
  1743.     EasyRequestArgs( window, &es, NULL, (APTR)(&dummyarg) );
  1744. }
  1745.  
  1746. void do_quit( struct IntuiMessage *imsg )
  1747. {
  1748.     running = FALSE;
  1749.     imsg->IDCMPWindow = NULL;               /* end NextSelect chain             */
  1750. }
  1751.  
  1752. void do_clear( struct IntuiMessage *imsg )
  1753. {
  1754.     memset( ¤t_pattern->dpGrid, 0, sizeof current_pattern->dpGrid );
  1755.     render_all();
  1756. }
  1757.  
  1758. /* ============================================================================ *
  1759.                                   Playback routines
  1760.  * ============================================================================ */
  1761.  
  1762.     /*  This calculates how big, in clock terms, the columns and the overall
  1763.         pattern is.
  1764.     */
  1765.  
  1766. void calc_pattern_ticks( void )
  1767. {
  1768.         /* calculate number of metric clocks per column */
  1769.  
  1770.     mclocks_per_column = ((MCLOCKS_PER_QNOTE * 4) >> drum_resolution);
  1771.  
  1772.         /* calculate the number of metric clocks per pattern */
  1773.  
  1774.     pattern_mclocks = total_columns * mclocks_per_column;
  1775. }
  1776.  
  1777.     /*  Calculate the tempo multiplication factor. The tempo is specified
  1778.         in quarter notes per minute; We need to convert that to metric
  1779.         clocks per real clock, plus a scaling factor of 256 (for 8-bit fixed
  1780.         point accuracy).
  1781.     */
  1782.  
  1783. void calc_tempo( void )
  1784. {   tempo_rate = current_tempo * (MCLOCKS_PER_QNOTE << 8) / (TICK_FREQ * 60);
  1785. }
  1786.  
  1787.     /*  Locate to a particular metric time (in this case "locating" is fairly
  1788.         trivial, since all patterns are required to be the same length.
  1789.     */
  1790.  
  1791. void locate_to_mclock( long mclocks )
  1792. {   WORD                column;
  1793.  
  1794.         /*  Unlike a normal division, we want numbers less than zero to
  1795.             round down, not up. So a seperate case is required for negative
  1796.             clock times.
  1797.         */
  1798.  
  1799.     mclock_accumulator = mclocks << 8;
  1800.  
  1801.     clock_pos = (mclock_accumulator / tempo_rate);
  1802.  
  1803.     if (mclocks >= 0)
  1804.     {   pattern_pos = mclocks / pattern_mclocks;
  1805.     }
  1806.     else
  1807.     {   pattern_pos = (mclocks - pattern_mclocks + 1) / pattern_mclocks;
  1808.     }
  1809.  
  1810.         /*  Calculate the start and end of the current pattern in metric ticks */
  1811.  
  1812.     current_pattern_start = pattern_pos * pattern_mclocks;
  1813.     current_pattern_end   = current_pattern_start + pattern_mclocks;
  1814.  
  1815.         /*  Now, figure out which column we are on */
  1816.  
  1817.     column = (mclocks - current_pattern_start) / mclocks_per_column;
  1818.  
  1819. /*  if (REVERSE) column_pos = total_columns - (column + 1) */
  1820.  
  1821.     column_pos = column;
  1822. }
  1823.  
  1824.     /*  Locate to a particular real time. Since we assume that tempo is
  1825.         constant (no tempo changes) this is also trivial -- but needs to be
  1826.         done in extended precision if we are going to allow long songs
  1827.         without overflow.
  1828.     */
  1829.  
  1830. void locate_to_tick ( long tick_count )
  1831. {
  1832.         /* REM: do extended multiply in assy... */
  1833.  
  1834.     locate_to_mclock( (tick_count * tempo_rate) >> 8 );
  1835. }
  1836.  
  1837. #if 0
  1838.     /*  Play to a particular clock time. This is different than the above
  1839.         routine in that it assumes that the current clock position is
  1840.         less than but near to the new clock time, and thus calculates
  1841.         using incremental addition rather than a full divide. If we were
  1842.         doing a full-on sequencer, we would exploit this philosophy even
  1843.         more fully.
  1844.  
  1845.         Also, it signals the tasks if the column position changed.
  1846.         Thus, the task only needs to wake up when a drum beat needs to
  1847.         actually be played.
  1848.     */
  1849.  
  1850. void play_to_mclock( long new_mclocks )
  1851. {   WORD                column;
  1852.  
  1853.         /*  See if the clock counted past the end of the frame */
  1854.  
  1855.     while (new_mclocks >= current_pattern_end)
  1856.     {   pattern_pos++;
  1857.         current_pattern_start = current_pattern_end;
  1858.         current_pattern_end   = current_pattern_start + pattern_mclocks;
  1859.         /* mark for signal...? */
  1860.     }
  1861.  
  1862.     column = (clock_count - current_pattern_start) / mclocks_per_column;
  1863.  
  1864. /*  if (REVERSE) column = total_columns - (column + 1) */
  1865.  
  1866.     if (column != column_pos)
  1867.     {   column_pos = column;
  1868.     }
  1869. }
  1870. #endif
  1871.